<template>
<div style="padding:6px;">
    <el-card v-show="showSearch" style="margin-bottom:6px;">
        <el-form :model="queryParams" ref="queryForm" :inline="true" label-width="68px" style="margin-bottom:-20px;">
            <el-form-item label="名称" prop="templateName">
                <el-input v-model="queryParams.templateName" placeholder="请输入物模型名称" clearable size="small" @keyup.enter.native="handleQuery" />
            </el-form-item>
            <el-form-item label="类别" prop="type">
                <el-select v-model="queryParams.type" placeholder="请选择模型类别" clearable size="small">
                    <el-option v-for="dict in dict.type.iot_things_type" :key="dict.value" :label="dict.label" :value="dict.value" />
                </el-select>
            </el-form-item>
            <el-form-item>
                <el-button type="primary" icon="el-icon-search" size="mini" @click="handleQuery">搜索</el-button>
                <el-button icon="el-icon-refresh" size="mini" @click="resetQuery">重置</el-button>
            </el-form-item>
        </el-form>
    </el-card>

    <el-card style="padding-bottom:100px;">
        <el-row :gutter="10" class="mb8">
            <el-col :span="1.5">
                <el-button type="primary" plain icon="el-icon-plus" size="mini" @click="handleAdd" v-hasPermi="['iot:template:add']">新增</el-button>
            </el-col>
            <el-col :span="1.5">
                <el-button type="success" plain icon="el-icon-edit" size="mini" :disabled="single" @click="handleUpdate" v-hasPermi="['iot:template:edit']">修改</el-button>
            </el-col>
            <el-col :span="1.5">
                <el-button type="danger" plain icon="el-icon-delete" size="mini" :disabled="multiple" @click="handleDelete" v-hasPermi="['iot:template:remove']">删除</el-button>
            </el-col>
            <el-col :span="1.5">
                <el-button type="warning" plain icon="el-icon-download" size="mini" @click="handleExport" v-hasPermi="['iot:template:export']">导出</el-button>
            </el-col>
            <right-toolbar :showSearch.sync="showSearch" @queryTable="getList"></right-toolbar>
        </el-row>

        <el-table v-loading="loading" :data="templateList" @selection-change="handleSelectionChange" border>
            <el-table-column type="selection" width="55" align="center" />
            <el-table-column label="名称" align="center" prop="templateName" />
            <el-table-column label="标识符" align="center" prop="identifier" />
            <el-table-column label="首页显示" align="center" prop="isTop" width="80">
                <template slot-scope="scope">
                    <dict-tag :options="dict.type.iot_yes_no" :value="scope.row.isTop" />
                </template>
            </el-table-column>
            <el-table-column label="监测值" align="center" prop="isMonitor" width="80">
                <template slot-scope="scope">
                    <dict-tag :options="dict.type.iot_yes_no" :value="scope.row.isMonitor" />
                </template>
            </el-table-column>
            <el-table-column label="系统定义" align="center" prop="isSys" width="80">
                <template slot-scope="scope">
                    <dict-tag :options="dict.type.iot_yes_no" :value="scope.row.isSys" />
                </template>
            </el-table-column>
            <el-table-column label="物模型类别" align="center" prop="type">
                <template slot-scope="scope">
                    <dict-tag :options="dict.type.iot_things_type" :value="scope.row.type" />
                </template>
            </el-table-column>
            <el-table-column label="数据类型" align="center" prop="datatype">
                <template slot-scope="scope">
                    <dict-tag :options="dict.type.iot_data_type" :value="scope.row.datatype" />
                </template>
            </el-table-column>
            <el-table-column label="数据定义" align="left" header-align="center" prop="specs" min-width="150" class-name="specsColor">
                <template slot-scope="scope">
                    <div v-html="formatSpecsDisplay(scope.row.specs)"></div>
                </template>
            </el-table-column>
            <el-table-column label="创建时间" align="center" prop="createTime" width="180">
                <template slot-scope="scope">
                    <span>{{ parseTime(scope.row.createTime, "{y}-{m}-{d}") }}</span>
                </template>
            </el-table-column>
            <el-table-column label="操作" align="center" class-name="small-padding fixed-width" width="150">
                <template slot-scope="scope">
                    <el-button size="small" type="primary" style="padding:5px;" icon="el-icon-edit" @click="handleUpdate(scope.row)" v-hasPermi="['iot:template:edit']">修改</el-button>
                    <el-button size="small" type="danger" style="padding:5px;" icon="el-icon-delete" @click="handleDelete(scope.row)" v-hasPermi="['iot:template:remove']" v-if=" canEdit ? true : (scope.row.isSys == '1' ? false : true)">删除</el-button>
                </template>
            </el-table-column>
        </el-table>

        <pagination v-show="total > 0" :total="total" :page.sync="queryParams.pageNum" :limit.sync="queryParams.pageSize" @pagination="getList" />

        <!-- 添加或修改通用物模型对话框 -->
        <el-dialog :title="title" :visible.sync="open" width="600px" append-to-body>
            <el-form ref="form" :model="form" :rules="rules" label-width="100px">
                <el-form-item label="名称" prop="templateName">
                    <el-input v-model="form.templateName" placeholder="请输入物模型名称，例如：温度" />
                </el-form-item>
                <el-form-item label="标识符" prop="identifier">
                    <el-input v-model="form.identifier" placeholder="请输入标识符，例如：temperature" />
                </el-form-item>
                <el-form-item label="模型类别" prop="type">
                    <el-radio-group v-model="form.type" @change="typeChange(form.type)">
                        <el-radio-button label="1">属性</el-radio-button>
                        <el-radio-button label="2">功能</el-radio-button>
                        <el-radio-button label="3">事件</el-radio-button>
                    </el-radio-group>
                </el-form-item>
                <el-form-item label="首页显示" prop="isTop" v-show="form.type != 3">
                    <el-switch v-model="form.isTop" active-text="" inactive-text="" :active-value="1" :inactive-value="0" active-color="#13ce66">
                    </el-switch>
                </el-form-item>
                <el-form-item label="实时监测" prop="isMonitor" v-show="form.type == 1">
                    <el-switch v-model="form.isMonitor" active-text="" inactive-text="" :active-value="1" :inactive-value="0" active-color="#13ce66" @change="changeMonitor(form.isMonitor)">
                    </el-switch>
                </el-form-item>
                <el-divider></el-divider>
                <el-form-item label="数据类型" prop="datatype">
                    <el-select v-model="form.datatype" placeholder="请选择数据类型" @change="dataTypeChange">
                        <el-option key="integer" label="整数" value="integer"></el-option>
                        <el-option key="decimal" label="小数" value="decimal"></el-option>
                        <el-option key="bool" label="布尔" value="bool" :disabled="form.isMonitor==1"></el-option>
                        <el-option key="enum" label="枚举" value="enum" :disabled="form.isMonitor==1"></el-option>
                        <el-option key="string" label="字符串" value="string" :disabled="form.isMonitor==1"></el-option>
                        <el-option key="array" label="数组" value="array" :disabled="form.isMonitor==1"></el-option>
                    </el-select>
                </el-form-item>
                <div v-if="form.datatype == 'integer' || form.datatype == 'decimal'">
                    <el-form-item label="取值范围">
                        <el-row>
                            <el-col :span="9">
                                <el-input v-model="form.specs.min" placeholder="最小值" type="number" />
                            </el-col>
                            <el-col :span="2" align="center">到</el-col>
                            <el-col :span="9">
                                <el-input v-model="form.specs.max" placeholder="最大值" type="number" />
                            </el-col>
                        </el-row>
                    </el-form-item>
                    <el-form-item label="单位">
                        <el-input v-model="form.specs.unit" placeholder="请输入单位，例如：℃" />
                    </el-form-item>
                    <el-form-item label="步长">
                        <el-input v-model="form.specs.step" placeholder="请输入步长，例如：1" type="number" />
                    </el-form-item>
                </div>
                <div v-if="form.datatype == 'bool'">
                    <el-form-item label="布尔值" prop="">
                        <el-row style="margin-bottom:10px;">
                            <el-col :span="11">
                                <el-input v-model="form.specs.falseText" placeholder="0 对应的文本，例如：关闭" />
                            </el-col>
                            <el-col :span="10" :offset="1"> 0 对应文本</el-col>
                        </el-row>
                        <el-row>
                            <el-col :span="11">
                                <el-input v-model="form.specs.trueText" placeholder="1 对应的文本，例如：打开" />
                            </el-col>
                            <el-col :span="10" :offset="1"> 1 对应文本</el-col>
                        </el-row>
                    </el-form-item>
                </div>
                <div v-if="form.datatype == 'enum'">
                    <el-form-item label="枚举项" prop="">
                        <el-row v-for="(item,index) in form.specs.enumList" :key="index" style="margin-bottom:10px;">
                            <el-col :span="8">
                                <el-input v-model="item.value" placeholder="参数值，例如：0" type="number" />
                            </el-col>
                            <el-col :span="12" :offset="1">
                                <el-input v-model="item.text" placeholder="参数描述，例如：中速档位" />
                            </el-col>
                            <el-col :span="2" :offset="1" v-if="index!=0"><a style="color:#F56C6C" @click="removeEnumItem(index)">删除</a></el-col>
                        </el-row>
                        <div>+ <a style="color:#409EFF" @click="addEnumItem()">添加枚举项</a></div>
                    </el-form-item>
                </div>
                <div v-if="form.datatype == 'string'">
                    <el-form-item label="最大长度" prop="">
                        <el-input v-model="form.specs.maxLength" placeholder="请输入字符串最大长度，例如：1024" type="number" />
                    </el-form-item>
                </div>
                <div v-if="form.datatype == 'array'">
                    <el-form-item label="数组类型" prop="">
                        <el-radio-group v-model="form.specs.arrayType">
                            <el-radio label="int">int（整数）</el-radio>
                            <el-radio label="double">double（小数）</el-radio>
                            <el-radio label="string">string（字符串）</el-radio>
                        </el-radio-group>
                    </el-form-item>
                </div>
            </el-form>

            <div slot="footer" class="dialog-footer">
                <el-button type="primary" @click="submitForm" :disabled=" canEdit ? false : (form.isSys == '1' ? true : false)">确 定</el-button>
                <el-button @click="cancel">取 消</el-button>
            </div>
        </el-dialog>

    </el-card>
</div>
</template>

<style>
.specsColor {
    background-color: #fcfcfc;
}
</style>

<script>
import {
    listTemplate,
    getTemplate,
    delTemplate,
    addTemplate,
    updateTemplate,
} from "@/api/iot/template";
export default {
    name: "Template",
    dicts: ["iot_things_type", "iot_data_type", "iot_yes_no"],
    data() {
        return {
            // 是否具有修改权限 admin可以修改系统定义的通用物 其他不可以
            canEdit: false,
            // 遮罩层
            loading: true,
            // 选中数组
            ids: [],
            // 非单个禁用
            single: true,
            // 非多个禁用
            multiple: true,
            // 显示搜索条件
            showSearch: true,
            // 总条数
            total: 0,
            // 通用物模型表格数据
            templateList: [],
            // 弹出层标题
            title: "",
            // 是否显示弹出层
            open: false,
            // 查询参数
            queryParams: {
                pageNum: 1,
                pageSize: 10,
                templateName: null,
                type: null,
                // 增加属性
                tenantName: ''
            },
            // 表单参数
            form: {},
            // 表单校验
            rules: {
                templateName: [{
                    required: true,
                    message: "物模型名称不能为空",
                    trigger: "blur"
                }, ],
                identifier: [{
                    required: true,
                    message: "标识符，产品下唯一不能为空",
                    trigger: "blur",
                }, ],
                type: [{
                    required: true,
                    message: "模型类别不能为空",
                    trigger: "change"
                }, ],
                datatype: [{
                    required: true,
                    message: "数据类型不能为空",
                    trigger: "change"
                }, ],
                specs: [{
                    required: true,
                    message: "数据定义不能为空",
                    trigger: "blur"
                }, ],
            },
        };
    },
    created() {
        this.getList();
        this.init();
    },
    methods: {
        init() {
            if (this.$store.state.user.roles.indexOf("admin") !== -1) {
                this.canEdit = true
            }
        },
        /** 查询通用物模型列表 */
        getList() {
            this.loading = true;
            if (this.$store.state.user.roles.indexOf("admin") === -1) {
                this.queryParams.tenantName = this.$store.state.user.name
            }
            listTemplate(this.queryParams).then((response) => {
                this.templateList = response.rows;
                this.total = response.total;
                this.loading = false;
            });
        },
        // 取消按钮
        cancel() {
            this.open = false;
            this.reset();
        },
        // 表单重置
        reset() {
            this.form = {
                templateId: null,
                templateName: null,
                userId: null,
                userName: null,
                tenantId: null,
                tenantName: null,
                identifier: null,
                type: null,
                datatype: null,
                isSys: null,
                isTop: null,
                isMonitor: null,
                delFlag: null,
                createBy: null,
                createTime: null,
                updateBy: null,
                updateTime: null,
                remark: null,
                specs: null,
            };
            this.resetForm("form");
        },
        /** 搜索按钮操作 */
        handleQuery() {
            this.queryParams.pageNum = 1;
            this.getList();
        },
        /** 重置按钮操作 */
        resetQuery() {
            this.resetForm("queryForm");
            this.handleQuery();
        },
        // 多选框选中数据
        handleSelectionChange(selection) {
            this.ids = selection.map((item) => item.templateId);
            this.single = selection.length !== 1;
            this.multiple = !selection.length;
        },
        /** 新增按钮操作 */
        handleAdd() {
            this.reset();
            this.open = true;
            this.title = "添加通用物模型";
            this.form.type = 1;
            this.form.datatype = "integer"
            this.form.specs = {
                enumList: [],
                arrayType: "int"
            };
        },
        /** 修改按钮操作 */
        handleUpdate(row) {
            this.reset();
            const templateId = row.templateId || this.ids;
            getTemplate(templateId).then((response) => {
                this.form = response.data;
                this.open = true;
                this.title = "修改通用物模型";
                // Json转对象
                this.form.specs = JSON.parse(this.form.specs);
            });
        },
        /** 提交按钮 */
        submitForm() {
            this.$refs["form"].validate((valid) => {
                if (valid) {
                    if (this.form.templateId != null) {
                        // 格式化specs
                        this.form.specs = this.formatThingsSpecs();
                        if (this.form.type == 2) {
                            this.form.isMonitor = 0;
                        } else if (this.form.type == 3) {
                            this.form.isMonitor = 0;
                            this.form.isTop = 0;
                        }
                        // 添加通用物模型的修改者
                        this.form.updateBy = this.$store.state.user.name
                        updateTemplate(this.form).then((response) => {
                            this.$modal.msgSuccess("修改成功");
                            this.open = false;
                            this.getList();
                        });
                    } else {
                        // 格式化specs
                        this.form.specs = this.formatThingsSpecs();
                        if (this.form.type == 2) {
                            this.form.isMonitor = 0;
                        } else if (this.form.type == 3) {
                            this.form.isMonitor = 0;
                            this.form.isTop = 0;
                        }
                        // 添加通用物模型的创造者
                        this.form.createBy = this.$store.state.user.name
                        addTemplate(this.form).then((response) => {
                            this.$modal.msgSuccess("新增成功");
                            this.open = false;
                            this.getList();
                        });
                    }
                }
            });
        },
        /** 删除按钮操作 */
        handleDelete(row) {
            const templateIds = row.templateId || this.ids;
            this.$modal
                .confirm('是否确认删除通用物模型编号为"' + templateIds + '"的数据项？')
                .then(function () {
                    return delTemplate(templateIds);
                })
                .then(() => {
                    this.getList();
                    this.$modal.msgSuccess("删除成功");
                })
                .catch(() => {});
        },
        /** 导出按钮操作 */
        handleExport() {
            this.download(
                "iot/template/export", {
                    ...this.queryParams,
                },
                `template_${new Date().getTime()}.xlsx`
            );
        },
        // 类型改变
        typeChange(label) {
            if (label == 2 || label == 3) {
                this.form.isMonitor = 0;
            }
        },
        // 实时监测改变
        changeMonitor(isMonitor) {
            if (isMonitor == 1 && this.form.datatype != "integer" && this.form.datatype != "decimal") {
                this.form.datatype = "integer";
            }
        },
        // 格式化物模型
        formatThingsSpecs() {
            var data = {};
            data.type = this.form.datatype;
            if (this.form.datatype == "integer" || this.form.datatype == "decimal") {
                data.min = Number(this.form.specs.min);
                data.max = Number(this.form.specs.max);
                data.unit = this.form.specs.unit;
                data.step = Number(this.form.specs.step);
            } else if (this.form.datatype == "string") {
                data.maxLength = Number(this.form.specs.maxLength);
            } else if (this.form.datatype == "bool") {
                data.falseText = this.form.specs.falseText;
                data.trueText = this.form.specs.trueText;
            } else if (this.form.datatype == "array") {
                data.arrayType = this.form.specs.arrayType;
            } else if (this.form.datatype == "enum") {
                data.enumList = this.form.specs.enumList;
            }
            return JSON.stringify(data);
        },
        /** 切换为枚举项 */
        dataTypeChange(val) {
            if (val == "enum") {
                this.form.specs.enumList = [{
                    value: "",
                    text: ""
                }];
            }
        },
        /** 添加枚举项 */
        addEnumItem() {
            this.form.specs.enumList.push({
                value: "",
                text: ""
            })
        },
        /** 删除枚举项 */
        removeEnumItem(index) {
            this.form.specs.enumList.splice(index, 1);
        },
        /** 格式化显示数据定义 */
        formatSpecsDisplay(json) {
            let specs = JSON.parse(json);
            if (specs.type === "integer" || specs.type === "decimal") {
                return "<span style='width:50%;display:inline-block;'>最大值：<span style=\"color:#F56C6C\">" + specs.max +
                    "</span></span>最小值：<span style=\"color:#F56C6C\">" + specs.min +
                    "</span><br /><span style='width:50%;display:inline-block;'>步长：<span style=\"color:#F56C6C\">" + specs.step +
                    "</span></span>单位：<span style=\"color:#F56C6C\">" + specs.unit;
            } else if (specs.type === "string") {
                return "最大长度：<span style=\"color:#F56C6C\">" + specs.maxLength + "</span>";
            } else if (specs.type === "array") {
                return "数组类型：<span style=\"color:#F56C6C\">" + specs.arrayType + "</span>";
            } else if (specs.type === "enum") {
                let items = "";
                for (let i = 0; i < specs.enumList.length; i++) {
                    items = items + "<span style='width:50%;display:inline-block;'>" + specs.enumList[i].value + "：<span style='color:#F56C6C'>" + specs.enumList[i].text + "</span></span>"
                    if (i > 0 && i % 2 != 0) {
                        items = items + "<br />"
                    }
                }
                return items;
            } else if (specs.type === "bool") {
                return "<span style='width:50%;display:inline-block;'>0：<span style=\"color:#F56C6C\">" + specs.falseText +
                    "</span></span>1：<span style=\"color:#F56C6C\">" + specs.trueText
            }
        },

    },
};
</script>
